home *** CD-ROM | disk | FTP | other *** search
/ START Magazine / START VOL 4 NO 10.st / info_src.arc / IN.PAS < prev    next >
Encoding:
Pascal/Delphi Source File  |  1990-02-01  |  61.0 KB  |  1,636 lines

  1. {InfoBase ST by James W. Maki (c) Copyright 1990 by Antic Publishing, Inc.}
  2. {$M+}
  3. {$E+}
  4.  
  5. Program Input_Module;
  6.  
  7.       {$I A:GEMSUBS.PAS }
  8.       {$I A:AUXSUBS.PAS }
  9.  
  10.  Const
  11.       {$I B:MOD_CONS.PAS }
  12.  
  13.  Type
  14.       {$I B:MOD_TYPE.PAS }
  15.  
  16.  Var
  17.       {$I B:MOD_VAR.PAS }
  18.  
  19. {   **********************  External  **********************************   }
  20.   PROCEDURE HelpScreen ;
  21.     EXTERNAL ;
  22.  
  23.   procedure ClrHome ;
  24.      External ;
  25.      
  26.   procedure NewCursor(ScrMode : short_integer) ;
  27.      External ;
  28.      
  29.   procedure EraseCursor(ScrMode : short_integer) ;
  30.      External ;
  31.  
  32.   procedure CheckCurLoc(Var CurLoc : short_integer ;
  33.                         Var Current : ScrPtr ; 
  34.                             XPos, YPos, ScrMode : short_integer ) ;
  35.      External ;
  36.  
  37.   procedure DetCurRec(    D_CurRec : DataStorePtr ;
  38.                       Var CurRec   : DataStorePtr ; 
  39.                       Var Location : short_integer ) ;
  40.      External ;
  41.  
  42.   procedure GetChar(      CurRec  : ScrPtr ;
  43.                         D_CurRec  : DataPtr ; 
  44.                     Var Character : StrChar ;
  45.                         Position  : short_integer ) ;
  46.      External ;
  47.  
  48.   procedure GetStr(CurRec   : DataPtr ; Var DisplayStr : Str255 ;
  49.                    StartPos, Size : short_integer ) ;
  50.      External ;
  51.  
  52.   procedure CheckOverLap( NewRec : ScrPtr ; X, Y : short_integer ; 
  53.                          Var OverLap : boolean ) ;
  54.      External ;
  55.  
  56.   procedure ChangeMode( Var Mode, NewMode : short_integer ) ;
  57.      External ;
  58.  
  59.   procedure MenuOption ;
  60.      External ;
  61.  
  62.   procedure Select_Open( Var NewMode : short_integer ) ;
  63.      External ;
  64.      
  65.   procedure Select_Close ;
  66.      External ;
  67.  
  68.   procedure Select_Save ;
  69.      External ;
  70.      
  71.   procedure ExitProgram ;
  72.      External ;
  73.  
  74.   procedure Paint_Frame( x, y, w : short_integer ) ;
  75.      External ;
  76.  
  77.   procedure EraseARec( CurRec : ScrPtr ) ;
  78.      External ;
  79.  
  80.   procedure DrawAField( CurRec : ScrPtr ) ;
  81.      External ;
  82.  
  83.   procedure DrawScreen( CurRec : ScrPtr ) ;
  84.      External ;
  85.  
  86.   procedure DrawRecord(CurRec : DataPtr) ;
  87.      External ;
  88.  
  89.   procedure DrawDZ_In ;
  90.      External ;
  91.      
  92.   procedure DrawDZ_Out ;
  93.      External ;
  94.           
  95.   procedure ModifyStr(CurRec : DataPtr ; Location : short_integer ; 
  96.                       InChar : char) ;
  97.      External ;
  98.  
  99.   procedure DeleteChar(ScrRec  : ScrPtr ; DataRec : DataPtr ; 
  100.                        Loc     : short_integer ) ;
  101.      External ;
  102.  
  103.   procedure InsertChar(CurRec  : ScrPtr ; DataRec : DataPtr ; 
  104.                        NewChar : char ;   Loc     : short_integer ) ;
  105.      External ;
  106.      
  107.   procedure Select_Modify ;
  108.      External ;
  109.  
  110.   procedure Select_Enter ;
  111.      External ;
  112.  
  113.   procedure SelectSearch( Var NewMode : short_integer ) ;
  114.      External ;
  115.  
  116.   procedure SelectInput( Var NewMode : short_integer ) ;
  117.      External ;
  118.  
  119.   procedure SelectOutput( Var NewMode : short_integer ) ;
  120.      External ;
  121.  
  122.   procedure SelectSort( Var NewMode : short_integer ) ;
  123.      External ;
  124.  
  125.   procedure DeleteARec(CurRec : ScrPtr) ;
  126.      External ;
  127.      
  128.   procedure DS_DeleteARec(CurRec : DataPtr) ;
  129.      External ;
  130.  
  131.   procedure ClearRecord( CurRec : DataPtr ) ;
  132.      External ;
  133.  
  134.   procedure CreateDataRec(DataNum : short_integer) ;
  135.      External ;
  136.  
  137.   procedure Int_AddARec(Var FirstRec, CurRec, LastRec : IntPtr ; 
  138.                             Value : short_integer ) ;
  139.      External ;
  140.  
  141.   procedure IncrementRec(Var  CurRec : DataPtr ; Value : short_integer ;
  142.                               DrawFlag : boolean ) ;
  143.      External ;
  144.  
  145.   procedure GoToFirst( Var CurRec : DataPtr ; DrawFlag : boolean ) ;
  146.      External ;
  147.      
  148.   procedure GoToLast(Var CurRec : DataPtr ; DrawFlag : boolean) ;
  149.      External ;
  150.      
  151.   procedure FormatCheck( CurRec : DataPtr ) ;
  152.      External ;
  153.      
  154.   procedure AutoDate(    ScrRec : ScrPtr ; DataRec : DataPtr ;
  155.                      Var DateStr : Str255 ) ;
  156.      External ;
  157.  
  158.   procedure GetAscii(    Character : StrChar ;
  159.                      Var CharInt : short_integer) ;
  160.      External ;
  161.  
  162. {   ***********************  Routines **********************************   }
  163.  
  164. {  ********************************************************************
  165.      Input Info returns a string containing a description of the Data Type
  166.      of the current field (i.e., translates DataType to a description).
  167. ************************************************************************* }
  168.   procedure InputInfo(Var FormatStr : Str255)  ;
  169.  
  170.     var
  171.         i, Count  : byte ;
  172.         TypeStr   : Str20 ;
  173.  
  174.      begin
  175.        for i := $41 to $48 do
  176.            if chr(i) = S_CurrentRec[ScrNum]^.DataType then
  177.               begin
  178.                 case i of
  179.                     $41 : TypeStr := 'String' ;
  180.                     $42 : TypeStr := 'Boolean' ;
  181.                     $43 : TypeStr := 'Integer' ;
  182.                     $44 : TypeStr := 'Company' ;
  183.                     $45 : TypeStr := 'Real' ;
  184.                     $46 : TypeStr := 'Dollar' ;
  185.                     $47 : TypeStr := 'Date' ;
  186.                     $48 : TypeStr := 'Name' ;
  187.                 end ;
  188.                 i := $50 ;
  189.               end ;
  190.  
  191.        WriteV(FormatStr, ' DataType :', TypeStr:8, chr($7C):3, 
  192.                          'Record #':12, RecNo[DataNum]:5,
  193.                          ' of', TotalRec[DataNum]:5, chr($7C):3) ; 
  194.      end ;
  195.  
  196. { *************************************************************************
  197.      Search Info returns a string containing a description of the search
  198.      criteria for the current field (i.e., translates a short_integer 
  199.      to a formula description).
  200. ************************************************************************* }
  201.   procedure SearchInfo(    S_Int : short_integer ;
  202.                        Var S_Str : Str20) ;
  203.  
  204.      begin
  205.        Case S_Int of
  206.            1 : S_Str := '=' ;
  207.            2 : S_Str := '>' ;
  208.            3 : S_Str := '<' ;
  209.            4 : S_Str := '<>' ;
  210.            5 : S_Str := '<=' ;
  211.            6 : S_Str := '>=' ;
  212.        end ;
  213.      end ;
  214.  
  215. { *************************************************************************
  216.      Update Info Line creates a string to be displayed on the Info Line.
  217.      The information displayed depends upon the current mode and search
  218.      or sort criteria.
  219. ************************************************************************* }
  220.   procedure UpdateInfoLine ;
  221.  
  222.     var
  223.        MatchStr  : Str20 ;
  224.        SearchStr : Str255 ;
  225.        L_I,
  226.        L_SlidePos : long_integer ;
  227.  
  228.      begin
  229.        UpdateFlag := false ;
  230.        L_SlidePos := 1 ;
  231.        L_I := 1 ;
  232.        Case Mode of
  233.            1 : WriteV(FormatStr,' Cursor : Line', YCur:3,
  234.                                 '   Column', XCur:3) ;
  235.            2 : begin
  236.                  InputInfo(FormatStr) ;
  237.                  if TotalRec[DataNum] > 1 then
  238.                     L_SlidePos := (L_I * 1000 * (RecNo[DataNum] - 1)) DIV 
  239.                                   (TotalRec[DataNum] - 1) ;
  240.                end ;
  241.            3 : if SearchFlag then
  242.                   begin
  243.                     SearchInfo(C_CurRec^.Match, MatchStr) ;
  244.                     WriteV(FormatStr,
  245.                        '| F1 =  | F2 >  | F3 <  | F4 <> | F5 <= | F6 >= |',
  246.                        'Search Criteria :':18, MatchStr:3) ;
  247.                   end
  248.                else
  249.                   begin
  250.                     InputInfo(SearchStr) ;
  251.                     WriteV(FormatStr, SearchStr, 'Search Rec #':13,
  252.                                       F_RecNo[DataNum]:5, ' of ',
  253.                                       F_TotalRec[DataNum]:5) ;
  254.                     if F_TotalRec[DataNum] > 1 then
  255.                        L_SlidePos := (L_I * 1000 * (F_RecNo[DataNum] - 1)) 
  256.                                       DIV (F_TotalRec[DataNum] - 1) ;
  257.                   end ;
  258.            4 : if SortFlag then
  259.                   WriteV(FormatStr,' Sort Mode : Shift-UpArrow | Shift-DownArrow')
  260.                else
  261.                   WriteV(FormatStr,' Sorting -- Please Wait .......') ;
  262.            5 : begin
  263.                  WriteV(FormatStr, ' Report Design : Line', (YCur - 7):5,
  264.                                    '   Column ', XCur:5) ;
  265.                  if TotScrRec > 10 then
  266.                     L_SlidePos := (L_I * 1000 * PL_Offset) 
  267.                                     DIV (TotScrRec - 10) ;
  268.                end ;
  269.        end ;
  270.        WindInfo[WindNum] := FormatStr ;
  271.        Set_WInfo(WindNum, WindInfo[WindNum]) ;
  272.  
  273.        Wind_Set(WindNum, WF_VSlide, L_SlidePos, 0, 0, 0) ;
  274.      end ;
  275.  
  276. { *************************************************************************
  277.      Delete Record deletes the current data record and redraws the new
  278.      current record.
  279. ************************************************************************* }
  280.   procedure DeleteRecord ;
  281.  
  282.      begin 
  283.        EditFlag[ScrNum] := true ;
  284.        Hide_Mouse ;
  285.        DS_DeleteARec(D_CurrentRec[DataNum]) ;
  286.        ClrHome ;
  287.        DrawRecord(D_CurrentRec[DataNum]) ;
  288.        UpdateFlag := true ; ;
  289.        Show_Mouse ;
  290.      end ;
  291.  
  292. { *************************************************************************
  293.        Delete a Screen Info field from the current design screen.
  294. ************************************************************************* }
  295.      procedure DeleteScrRec ;
  296.  
  297.        var
  298.           CurLoc : short_integer ;
  299.           CurRec : ScrPtr ;
  300.           TotalOffset : short_integer ;
  301.  
  302.         begin
  303.           if S_CurrentRec[ScrNum] <> nil then
  304.              begin
  305.                CheckCurLoc( CurLoc, CurRec, XCur, YCur, ScrNum ) ;
  306.                if CurLoc > -1 then
  307.                   begin
  308.                     Hide_Mouse ;
  309.                     D_EditFlag[ScrNum] := true ;
  310.                     DeleteARec(CurRec) ;
  311.                   { Recalculate Offsets  }
  312.                     CurRec := S_FirstRec[ScrNum] ;
  313.                     TotalOffset := 0 ;
  314.                     While CurRec <> nil do
  315.                       begin
  316.                         CurRec^.Offset := TotalOffset ;
  317.                         TotalOffset := TotalOffset + CurRec^.Size ;
  318.                         CurRec := CurRec^.Next ;
  319.                       end ;
  320.                    DrawScreen(S_FirstRec[ScrNum]) ;
  321.                    Show_Mouse ;
  322.                  end ;
  323.              end ;
  324.         end ;
  325.  
  326. { *************************************************************************
  327.        CharInInput either inserts the character or modifies the string,
  328.        depending upon the cursor location, then displays the modified
  329.        string.
  330. ************************************************************************* }
  331.      procedure CharInInput(LoByte : short_integer) ;
  332.  
  333.        var
  334.            NewChar    : StrChar ;
  335.            Location   : short_integer ;
  336.            DisplayStr : Str255 ;
  337.  
  338.         begin
  339.           EditFlag[ScrNum] := true ;
  340.           if (S_CurrentRec[ScrNum]^.DataType = 'B') OR
  341.              (S_CurrentRec[ScrNum]^.DataType = 'D') then
  342.              begin
  343.                if (LoByte = $54) OR (LoByte = $74) then
  344.                   LoByte := $54
  345.                else
  346.                   if (LoByte = $46) OR (LoByte = $66) then
  347.                      LoByte := $46
  348.                   else
  349.                      if (LoByte <> $02) AND (LoByte <> $03) then
  350.                         LoByte := $01 ;
  351.              end ;
  352.  
  353.           if LoByte <> $01 then
  354.              begin
  355.                Location := S_CurrentRec[ScrNum]^.XInPos +
  356.                            S_CurrentRec[ScrNum]^.Offset ;
  357.                DetCurRec(D_CurrentRec[DataNum]^.Data, D_DataRec, Location) ;
  358.                GetChar(S_CurrentRec[ScrNum],D_CurrentRec[DataNum], NewChar, 
  359.                    S_CurrentRec[ScrNum]^.XInPos + S_CurrentRec[ScrNum]^.Offset) ;
  360.                Location := S_CurrentRec[ScrNum]^.XInPos +
  361.                            S_CurrentRec[ScrNum]^.Offset ;
  362.                if NewChar = chr(1) then
  363.                   ModifyStr(D_CurrentRec[DataNum], Location, chr(LoByte))
  364.                else
  365.                   InsertChar(S_CurrentRec[ScrNum], D_CurrentRec[DataNum], 
  366.                              chr(LoByte), S_CurrentRec[ScrNum]^.XInPos + 1) ;
  367.                GetStr(D_CurrentRec[DataNum], DisplayStr, 
  368.                       S_CurrentRec[DataNum]^.Offset, 
  369.                       S_CurrentRec[DataNum]^.Size ) ;
  370.                       
  371.                Hide_Mouse ;
  372.                Draw_String(x + S_CurrentRec[ScrNum]^.XPos * 8,
  373.                            y + YCur * Spacing, DisplayStr) ;
  374.                Show_Mouse ;
  375.  
  376.                if S_CurrentRec[ScrNum]^.XInPos + 1 < 
  377.                   S_CurrentRec[ScrNum]^.Size then
  378.                   begin
  379.                     XCur := XCur + 1 ;
  380.                     S_CurrentRec[ScrNum]^.XInPos := 
  381.                       S_CurrentRec[ScrNum]^.XInPos + 1 ; 
  382.                   end ;
  383.              end ;
  384.         end ;
  385. { *************************************************************************
  386.      DrawGetStr will correctly position the output strings to the scrolling
  387.      Report Design screen.  The string will be correctly modified for
  388.      the field strings which are displayed as ASCII but stored as modified
  389.      ASCII to differential during report formatting.
  390. ************************************************************************* }
  391.   procedure DrawGetStr ;
  392.  
  393.     var
  394.        i         : short_integer ;
  395.        CheckChar : char ;
  396.  
  397.      begin
  398.        Hide_Mouse ;
  399.        if RW_Offset > 0 then
  400.           GetStr(D_CurrentRec[Report], FormatStr,
  401.                  S_CurrentRec[Report]^.Offset + 56, 76)
  402.        else
  403.           GetStr(D_CurrentRec[Report], FormatStr,
  404.                  S_CurrentRec[Report]^.Offset, 80);
  405.  
  406.        for i := 1 to Length(Formatstr) do
  407.            begin
  408.              CheckChar := FormatStr[i] ;
  409.              if ord(CheckChar) > $7F then
  410.                 FormatStr[i] := chr(ord(CheckChar) - $80 + $41) ;
  411.            end ;
  412.  
  413.        if RW_Offset > 0 then
  414.           Draw_String(x, y + YCur * Spacing - 4 * Resolution, FormatStr)
  415.        else
  416.           Draw_String(x + 8, y + YCur * Spacing - 4 * Resolution, FormatStr) ;
  417.        Show_Mouse ;
  418.      end ;
  419.  
  420. { *************************************************************************
  421.        CharRPInput either inserts the character or modifies the string,
  422.        depending upon the cursor location, then displays the modified
  423.        string.
  424. ************************************************************************* }
  425.      procedure CharRPInput(LoByte : short_integer) ;
  426.  
  427.        var
  428.            Location : short_integer ;
  429.            NewChar  : StrChar ;
  430.  
  431.         begin
  432.           R_EditFlag := true ;
  433.           InsertChar(S_CurrentRec[Report], D_CurrentRec[Report], 
  434.                      chr(LoByte), XCur) ;
  435.           DrawGetStr ;
  436.           XCur := XCur + 1 ;
  437.           UpdateFlag := true ; ;
  438.         end ;
  439.  
  440. { *************************************************************************
  441.        Creates a new DataStore Rec at the end of the current list and
  442.        makes the new record the current record.  For adding new information
  443.        to the database.
  444. ************************************************************************* }
  445.      procedure Press_Tab ;
  446.      
  447.         begin
  448.           if NOT FullMemory then
  449.              begin
  450.                EditFlag[ScrNum] := true ;
  451.                FormatCheck(D_CurrentRec[DataNum]) ;
  452.                ClearRecord(D_CurrentRec[DataNum]) ;
  453.                CreateDataRec(DataNum) ;
  454.                RecNo[DataNum] := TotalRec[DataNum] ;
  455.              end
  456.           else
  457.              begin
  458.                AlertStr := '[2][-Memory is Full- | |' ;
  459.                AlertStr := Concat(AlertStr, 'You May Not Add | |') ;
  460.                AlertStr := Concat(AlertStr, 'Any More Records]') ;
  461.                AlertStr := Concat(AlertStr, '[ Continue ]') ;
  462.                Result   := Do_Alert(AlertStr,1) ;
  463.              end ;
  464.           UpdateFlag := true ; ;
  465.         end ;
  466.  
  467. { ********************** Keyboard Input Routines ************************* }
  468. { *************************************************************************
  469.      This procedure is called by the procedure Keyboard_Input module
  470.      below. Event_Loop passes the variable KeyValue, the keyboard scan
  471.      code, to Keyboard_Input.  KB_InReport seperates KeyValue
  472.      into HiByte and LoByte components and then scans the values to 
  473.      determine the outcome.  The old cursor position is cleared at the 
  474.      beginning of the procedure and the new cursor position is drawn
  475.      at the end of the procedure.  KB_InReport is really a cursor control
  476.      procedure.
  477. ************************************************************************* }
  478.   procedure KB_InReport( HiByte, LoByte, NewMode : short_integer ;
  479.                         Var KeyIn : boolean ) ;
  480.  
  481.     var
  482.        i,
  483.        CurLoc,
  484.        Location,
  485.        TotalOffset,
  486.        SaveCur     : short_integer ;
  487.        CurRec      : ScrPtr ;
  488.        ReDrawScr   : boolean ;
  489.  
  490.      begin
  491.        ReDrawScr := false ;
  492.  
  493.        if LoByte = $00 then 
  494.           Case HiByte of
  495.               $3B : begin
  496.                       PL_Offset := PL_Offset - 10 ;
  497.                       if PL_Offset < 0 then PL_Offset := 0 ;
  498.                       DrawDZ_In ;
  499.                     end ;
  500.               $3C : begin
  501.                       PL_Offset := PL_Offset + 10 ;
  502.                       if PL_Offset + 10 > TotScrRec then
  503.                          PL_Offset := TotScrRec - 9 ;
  504.                       if PL_Offset < 0 then PL_Offset := 0 ;
  505.                       DrawDZ_In ;
  506.                     end ;
  507.                     
  508.               $47 : begin                  { Clr Home }
  509.                       XCur := 1 ;
  510.                       YCur := 8 ;
  511.                       RW_Offset := 0 ;
  512.                       S_CurrentRec[Report] := S_FirstRec[Report] ;
  513.                       DrawDZ_Out ;
  514.                     end ;
  515.               $48 : if YCur > 8 then
  516.                        begin
  517.                          KeyIn := true ;
  518.                          YCur := YCur - 1 ;  { up }
  519.                          S_CurrentRec[Report] := S_CurrentRec[Report]^.Prev ;
  520.                        end ;
  521.               $4B : begin
  522.                       KeyIn := true ;
  523.                       if XCur - RW_Offset > 1 then
  524.                          XCur := XCur - 1    { left }
  525.                       else
  526.                          if XCur > 1 then
  527.                             begin
  528.                               XCur := XCur - 1 ;
  529.                               RW_Offset := 0 ;
  530.                               ReDrawScr := true ;
  531.                             end ;
  532.                     end ;
  533.               $4D : begin
  534.                       KeyIn := true ;
  535.                       if (XCur - RW_Offset < 75) AND
  536.                          (XCur < RepWidth) then
  537.                           XCur := XCur + 1    { right }
  538.                       else
  539.                          if XCur < RepWidth - 1 then
  540.                             begin
  541.                               XCur := XCur + 1 ;
  542.                               RW_Offset := 57 ;
  543.                               ReDrawScr := true ;
  544.                             end ;
  545.                     end ;
  546.               $50 : if ((YCur < h DIV Spacing) AND (P_Mode = 2)) OR 
  547.                        ((YCur < 7 + LabLine) AND (P_Mode <> 2))  then
  548.                        begin
  549.                          KeyIn := true ;
  550.                          YCur := YCur + 1 ;  { down }
  551.                          S_CurrentRec[Report] := S_CurrentRec[Report]^.Next ;
  552.                        end ;
  553.           end 
  554.        else
  555.   
  556.           if (HiByte = $01) AND (LoByte = $1B) then           { ESC }
  557.              begin
  558.                R_EditFlag := true ;
  559.                for i := 0 to 131 do
  560.                    begin
  561.                      Location := S_CurrentRec[Report]^.Offset + i ;
  562.                      ModifyStr(D_CurrentRec[Report], Location, chr($20)) ;
  563.                    end ;
  564.                DrawGetStr ;
  565.              end 
  566.           else
  567.  
  568.          if (HiByte = $0E) AND (LoByte = $08) then      { backspace }
  569.             begin
  570.               SaveCur := XCur ;
  571.               if XCur - RW_Offset > 1 then
  572.                  XCur := XCur - 1
  573.               else
  574.                  if XCur > 1 then
  575.                     begin
  576.                       XCur := XCur - 1 ;
  577.                       RW_Offset := 0 ;
  578.                       ReDrawScr := true ;
  579.                     end ;
  580.               if XCur < SaveCur then
  581.                  begin
  582.                    R_EditFlag := true ;
  583.                    Location := XCur ;
  584.                    DeleteChar(S_CurrentRec[Report], D_CurrentRec[Report], 
  585.                               Location ) ;
  586.                    DrawGetStr ;
  587.                  end ;
  588.             end 
  589.           else
  590.  
  591.           if (HiByte = $53) AND (LoByte = $7F) then           { delete }
  592.              begin
  593.                R_EditFlag := true ;
  594.                Location := XCur ;
  595.                DeleteChar(S_CurrentRec[Report], D_CurrentRec[Report], Location ) ;
  596.                DrawGetStr ;
  597.              end
  598.           else
  599.  
  600.           if ((HiByte = $1C) AND (LoByte = $0D) OR
  601.               (HiByte = $72) AND (LoByte = $0D)) AND         { return }
  602.               (((YCur < h DIV Spacing) AND (P_Mode = 2)) OR 
  603.               ((YCur < 7 + LabLine) AND (P_Mode <> 2)))  then
  604.               begin
  605.                 KeyIn := true ;
  606.                 YCur := YCur + 1 ;
  607.                 XCur := 1 ;
  608.                 if RW_Offset > 0 then
  609.                    begin
  610.                      RW_Offset := 0 ;
  611.                      ReDrawScr := true ;
  612.                    end ;
  613.                 S_CurrentRec[Report] := S_CurrentRec[Report]^.Next ;
  614.               end ;
  615.  
  616.        if ReDrawScr then
  617.           DrawDZ_Out ;
  618.        UpdateFlag := true ; ;
  619.      end;
  620.  
  621. { *************************************************************************
  622.      This procedure is called by the procedure Keyboard_Input module
  623.      below. Event_Loop passes the variable KeyValue, the keyboard scan
  624.      code, to Keyboard_Input.  KB_InDesign seperates KeyValue
  625.      into HiByte and LoByte components and then scans the values to 
  626.      determine the outcome.  The old cursor position is cleared at the 
  627.      beginning of the procedure and the new cursor position is drawn
  628.      at the end of the procedure.  KB_InDesign is really a cursor control
  629.      procedure.
  630. ************************************************************************* }
  631.   procedure KB_InDesign( HiByte, LoByte, NewMode : short_integer ;
  632.                         Var KeyIn : boolean ) ;
  633.     var
  634.        Dummy,
  635.        CurLoc,
  636.        TotalOffset : short_integer ;
  637.        CurRec      : ScrPtr ; 
  638.        OverLap     : Boolean ;
  639.  
  640. { *************************************************************************
  641.      Modified pertinent variables for a cursor change ;
  642. ************************************************************************* }
  643.      procedure ChangeCurPos(    CurRec : ScrPtr ; XFlag : Boolean ;
  644.                             Var XY, Pos, InPos, Cur : short_integer ;
  645.                                 Value : short_integer) ;
  646.      
  647.         begin
  648.           D_EditFlag[ScrNum] := true ;
  649.           XY := XY + Value ;
  650.           Pos := Pos + Value ;
  651.           if XFlag then
  652.              InPos := InPos + Value ;
  653.           Cur := Cur + Value ;
  654.           DrawAField(CurRec) ;
  655.         end ;
  656. { *************************************************************************
  657.        Move the current Screen Info field up one line -- if possible.
  658. ************************************************************************* }
  659.      procedure ShiftUp ;
  660.  
  661.         begin
  662.           CheckCurLoc( CurLoc, CurRec, XCur, YCur, ScrNum ) ;
  663.           if CurLoc > -1 then
  664.              if (CurRec^.Y > 1) AND (CurRec^.DataType <> 'D') then
  665.                 begin
  666.                   CheckOverLap(CurRec, CurRec^.X, CurRec^.Y - 1, OverLap) ;
  667.                   if NOT OverLap then
  668.                      begin
  669.                        EraseARec(CurRec) ;
  670.                        ChangeCurPos(CurRec, false, CurRec^.Y, CurRec^.YPos,
  671.                                     CurRec^.XInPos, YCur, -1) ;
  672.                        if CurRec^.DataType = 'H' then
  673.                           ChangeCurPos(CurRec^.Next, false, CurRec^.Next^.Y, 
  674.                                        CurRec^.Next^.YPos, 
  675.                                        CurRec^.Next^.XInPos, Dummy, -1) ;
  676.                      end ;
  677.                 end ;
  678.         end ;
  679. { *************************************************************************
  680.        Move the current Screen Info field one space left -- if possible.
  681. ************************************************************************* }
  682.      procedure ShiftLeft ;
  683.      
  684.         begin
  685.           CheckCurLoc( CurLoc, CurRec, XCur, YCur, ScrNum ) ;
  686.           if CurLoc > -1 then
  687.              if (CurRec^.X > 1) AND (CurRec^.DataType <> 'D') then
  688.                 begin
  689.                   CheckOverLap(CurRec, CurRec^.X - 1, CurRec^.Y, OverLap) ;
  690.                   if NOT OverLap then
  691.                      begin
  692.                        EraseARec(CurRec) ;
  693.                        ChangeCurPos(CurRec, true, CurRec^.X, CurRec^.XPos,
  694.                                     CurRec^.XInPos, XCur, -1) ;
  695.                        if CurRec^.DataType = 'H' then
  696.                           ChangeCurPos(CurRec^.Next, false, CurRec^.Next^.X, 
  697.                                        CurRec^.Next^.XPos, 
  698.                                        CurRec^.Next^.XInPos, Dummy, -1) ;
  699.                      end ;
  700.                 end ; 
  701.         end ;
  702. { *************************************************************************
  703.        Move the current Screen Info field one space right -- if possible.
  704. ************************************************************************* }
  705.      procedure ShiftRight ;
  706.      
  707.         begin
  708.           CheckCurLoc( CurLoc, CurRec, XCur, YCur, ScrNum ) ;
  709.           if CurLoc > -1 then
  710.              if (CurRec^.X + Length(CurRec^.LabelStr) + CurRec^.Size < 
  711.                  w DIV 8 - 5) AND (CurRec^.DataType <> 'D') then
  712.                 begin
  713.                   CheckOverLap(CurRec, CurRec^.X + 1, CurRec^.Y, OverLap) ;
  714.                   if NOT OverLap then
  715.                      begin
  716.                        EraseARec(CurRec) ;
  717.                        ChangeCurPos(CurRec, true, CurRec^.X, CurRec^.XPos,
  718.                                     CurRec^.XInPos, XCur, 1) ;
  719.                        if CurRec^.DataType = 'H' then
  720.                           ChangeCurPos(CurRec^.Next, false, CurRec^.Next^.X, 
  721.                                        CurRec^.Next^.XPos, 
  722.                                        CurRec^.Next^.XInPos, Dummy, 1) ;
  723.                      end ;
  724.                 end ; 
  725.         end ;
  726. { *************************************************************************
  727.        Move the current Screen Info field down one line -- if possible.
  728. ************************************************************************* }
  729.      procedure ShiftDown ;
  730.      
  731.         begin
  732.           CheckCurLoc( CurLoc, CurRec, XCur, YCur, ScrNum ) ;
  733.           if CurLoc > -1 then
  734.              if (CurRec^.Y < h DIV Spacing) AND
  735.                 (CurRec^.DataType <> 'D') then
  736.                 begin
  737.                   CheckOverLap(CurRec, CurRec^.X, CurRec^.Y + 1, OverLap) ;
  738.                   if NOT OverLap then
  739.                      begin
  740.                        EraseARec(CurRec) ;
  741.                        ChangeCurPos(CurRec, false, CurRec^.Y, CurRec^.YPos,
  742.                                     CurRec^.XInPos, YCur, 1) ;
  743.                        if CurRec^.DataType = 'H' then
  744.                           ChangeCurPos(CurRec^.Next, false, CurRec^.Next^.Y, 
  745.                                        CurRec^.Next^.YPos, 
  746.                                        CurRec^.Next^.XInPos, Dummy, 1) ;
  747.                      end ;
  748.                 end ;
  749.         end ;
  750.  
  751.      begin
  752.        if LoByte = $00 then 
  753.           Case HiByte of
  754.               $47 : begin                  { Clr Home }
  755.                       XCur := 1 ;
  756.                       YCur := 1 ;
  757.                     end ;
  758.               $48 : if YCur > 1 then
  759.                        YCur := YCur - 1 ;  { up }
  760.               $4B : if XCur > 1 then
  761.                        XCur := XCur - 1 ;  { left }
  762.               $4D : if XCur < w DIV 8 - 2 then
  763.                        XCur := XCur + 1 ;  { right }
  764.               $50 : if YCur < h DIV Spacing  then
  765.                        YCur := YCur + 1 ;  { down }
  766.           end
  767.        else
  768.           Case HiByte of
  769. { ^E }       $12 : if LoByte = $05 then
  770.                       Select_Enter ;
  771. { ^D }       $20 : if LoByte = $04 then
  772.                       DeleteScrRec ;
  773. { ^M }       $32 : if LoByte = $0D then
  774.                       Select_Modify ;
  775.              $48 : if LoByte = $38 then  { Shift-Up }
  776.                       ShiftUp ;
  777.              $4B : if LoByte = $34 then  { Shift-Left }
  778.                       ShiftLeft ;
  779.              $4D : if LoByte = $36 then  { Shift-Right }
  780.                       ShiftRight ;
  781.              $50 : if LoByte = $32 then  { Shift-Down }
  782.                       ShiftDown ;
  783.           end ;
  784.        UpdateFlag := true ; ;
  785.      end;
  786.  
  787. { ********************** Keyboard Input Routines ************************* }
  788. { *************************************************************************
  789.      KB_InInput is called by the Keyboard_Input module below.  It handles
  790.      keyboard input for the Input phase of the program.  KeyValue is 
  791.      passed from Keyboard_Input and the code is separated into HiByte and
  792.      LoByte components.  Cursor control and character input are handled
  793.      by this module.
  794. ************************************************************************* }
  795.   procedure KB_InInput( HiByte, LoByte, NewMode : short_integer ;
  796.                        Var KeyIn : boolean ) ;
  797.     var
  798.        Location,
  799.        CurLoc   : short_integer ;
  800.        CurRec   : ScrPtr ; 
  801.        OverLap  : Boolean ;
  802.        DisplayStr : Str255 ;
  803.        NewChar  : StrChar ;
  804.  
  805. { *************************************************************************
  806.        Select Date automatically inserts the current system date into
  807.        a Date DataType field.
  808. ************************************************************************* }
  809.      procedure SelectDate(ScrRec : ScrPtr ; DataRec : DataPtr ; 
  810.                           DisplayStr : Str255 ) ;
  811.  
  812.        var
  813.           i : short_integer ;
  814.  
  815.         begin
  816.           if (ScrRec^.DataType = 'G') AND 
  817.              ((Mode = 2) OR (Mode = 3)) then
  818.              begin
  819.                EditFlag[ScrNum] := true ;
  820.                Hide_Mouse ;
  821.                AutoDate(ScrRec, DataRec, DisplayStr) ;
  822.                for i := 1 to ScrRec^.Size - Length(DisplayStr) do
  823.                    DisplayStr := Concat(DisplayStr, chr($20)) ;
  824.                Draw_String(x + ScrRec^.XPos * 8, 
  825.                            y + ScrRec^.YPos * Spacing, DisplayStr) ;
  826.                Show_Mouse ;
  827.              end ;
  828.         end ;
  829.  
  830. { *************************************************************************
  831.        Moves the Cursor to the next field, or to the first field if the
  832.        current field is the last field.
  833. ************************************************************************* }
  834.      procedure Press_Ret ;
  835.  
  836.         begin
  837.           KeyIn := true ;
  838.           if S_CurrentRec[ScrNum]^.Next <> nil then
  839.              S_CurrentRec[ScrNum] := S_CurrentRec[ScrNum]^.Next
  840.           else
  841.              S_CurrentRec[ScrNum] := S_FirstRec[ScrNum] ;
  842.              XCur := S_CurrentRec[ScrNum]^.XPos ;
  843.              YCur := S_CurrentRec[ScrNum]^.YPos ;
  844.              S_CurrentRec[ScrNum]^.XInPos := 0 ;
  845.              UpdateFlag := true ; ;
  846.         end ;
  847.  
  848. { *************************************************************************
  849.        Deletes the character at the current cursor location.
  850. ************************************************************************* }
  851.      procedure Press_Del ;
  852.  
  853.         begin
  854.           KeyIn := true ;
  855.           EditFlag[ScrNum] := true ;
  856.           Location := S_CurrentRec[ScrNum]^.XInPos + 1 ;
  857.           DeleteChar(S_CurrentRec[ScrNum], D_CurrentRec[DataNum], Location ) ;
  858.           GetStr(D_CurrentRec[DataNum], DisplayStr, 
  859.                  S_CurrentRec[ScrNum]^.Offset, S_CurrentRec[ScrNum]^.Size ) ;
  860.           DisplayStr := Concat(DisplayStr, chr($20)) ;
  861.           Hide_Mouse ;
  862.           Draw_String(x + S_CurrentRec[ScrNum]^.XPos * 8,
  863.                       y + YCur * Spacing, DisplayStr) ;
  864.           Show_Mouse ;
  865.         end ;
  866.  
  867. { *************************************************************************
  868.        Deletes the character to the left of the current cursor location.
  869. ************************************************************************* }
  870.      procedure Press_BS ;
  871.  
  872.         begin
  873.           KeyIn := true ;
  874.           EditFlag[ScrNum] := true ;
  875.           S_CurrentRec[ScrNum]^.XInPos := S_CurrentRec[ScrNum]^.XInPos - 1 ;
  876.           XCur := XCur - 1 ;
  877.                            
  878.           Location := S_CurrentRec[ScrNum]^.XInPos + 1 ;
  879.           DeleteChar(S_CurrentRec[ScrNum], D_CurrentRec[DataNum], Location ) ;
  880.           GetStr(D_CurrentRec[DataNum], DisplayStr,
  881.                  S_CurrentRec[ScrNum]^.Offset, S_CurrentRec[ScrNum]^.Size) ;
  882.           DisplayStr := Concat(DisplayStr, chr($20)) ;
  883.           Hide_Mouse ;
  884.           Draw_String(x + S_CurrentRec[ScrNum]^.XPos * 8,
  885.                       y + YCur * Spacing, DisplayStr) ;
  886.           Show_Mouse ;
  887.         end ;
  888.  
  889. { *************************************************************************
  890.        Clears the current field of all previous information.
  891. ************************************************************************* }
  892.      procedure Press_ESC ;
  893.  
  894.        var
  895.            i      : short_integer ;
  896.            ScrRec : ScrPtr ;
  897.            Start  : short_integer ;
  898.  
  899.         begin
  900.           KeyIn := true ;
  901.           EditFlag[ScrNum] := true ;
  902.           ScrRec := S_CurrentRec[ScrNum] ;
  903.  
  904.           if ScrRec^.DataType = 'F' then
  905.              Start := 1
  906.           else
  907.              Start := 0 ;
  908.           for i := Start to ScrRec^.Size - 1 do
  909.               begin
  910.                 Location := ScrRec^.Offset + i ;
  911.                 ModifyStr(D_CurrentRec[DataNum], Location, chr(1)) ;
  912.              end ;
  913.  
  914.           Paint_Frame(x + (ScrRec^.X + Length(ScrRec^.LabelStr) + 2) * 8 + 4, 
  915.                       y + (ScrRec^.Y - 1) * Spacing + (4 * Resolution), 
  916.                       ScrRec^.Size * 8 ) ;
  917.  
  918.           Hide_Mouse ;
  919.           if ScrRec^.DataType = 'F' then
  920.              Draw_String(x + ScrRec^.XPos * 8,
  921.                          y + ScrRec^.YPos * Spacing, chr($24)) ;
  922.           Show_Mouse ;
  923.  
  924.           XCur := ScrRec^.XPos ;
  925.           ScrRec^.XInPos := 0 ;
  926.         end ;
  927.  
  928. { *************************************************************************
  929.        Move the cursor to the previous Screen Info field.
  930. ************************************************************************* }
  931.      procedure UpArrow(Var CurRec : ScrPtr) ;
  932.  
  933.         begin
  934.           KeyIn := true ;
  935.           if CurRec^.Prev <> nil then
  936.              CurRec := CurRec^.Prev
  937.           else
  938.              CurRec := S_LastRec[ScrNum] ;
  939.  
  940.           if Mode = 3 then
  941.              begin
  942.                if C_CurRec^.Prev <> nil then
  943.                   C_CurRec := C_CurRec^.Prev
  944.                else
  945.                   C_CurRec := C_LastRec ;
  946.              end ;
  947.  
  948.           XCur := CurRec^.XPos ;
  949.           YCur := CurRec^.YPos ;
  950.           CurRec^.XInPos := 0 ;
  951.           UpdateFlag := true ; ;
  952.         end ;
  953.  
  954. { *************************************************************************
  955.        Move the cursor to the left one character.
  956. ************************************************************************* }
  957.      procedure LeftArrow(Var CurRec : ScrPtr) ;
  958.        
  959.         begin
  960.           KeyIn := true ;
  961.           if CurRec^.XInPos - 1 >= 0 then
  962.              begin
  963.                CurRec^.XInPos := CurRec^.XInPos - 1 ;
  964.                XCur := XCur - 1 ;
  965.              end ;
  966.         end ;
  967.  
  968. { *************************************************************************
  969.        Move the cursor to the rigth one character.
  970. ************************************************************************* }
  971.      procedure RightArrow(Var CurRec : ScrPtr) ;
  972.      
  973.         begin
  974.           KeyIn := true ;     { <= }
  975.           if CurRec^.XInPos + 1 < CurRec^.Size then
  976.              begin
  977.                GetChar(CurRec,D_CurrentRec[DataNum], NewChar, 
  978.                        CurRec^.XInPos + CurRec^.Offset) ;
  979.                if NewChar <> chr(1) then
  980.                   begin
  981.                     CurRec^.XInPos := CurRec^.XInPos + 1 ;
  982.                     XCur := XCur + 1 ;
  983.                   end ;
  984.              end ;
  985.         end ;
  986.  
  987. { *************************************************************************
  988.        Move the cursor to the next Screen Pointer field.
  989. ************************************************************************* }
  990.      procedure DownArrow(Var CurRec : ScrPtr) ;
  991.      
  992.         begin
  993.           KeyIn := true ;
  994.           if CurRec^.Next <> nil then
  995.              CurRec := CurRec^.Next
  996.           else
  997.              CurRec := S_FirstRec[ScrNum] ;
  998.  
  999.           if Mode = 3 then
  1000.              begin
  1001.                if C_CurRec^.Next <> nil then
  1002.                   C_CurRec := C_CurRec^.Next
  1003.                else
  1004.                   C_CurRec := C_FirstRec ; 
  1005.              end ;
  1006.  
  1007.           XCur := CurRec^.XPos ;
  1008.           YCur := CurRec^.YPos ;
  1009.           CurRec^.XInPos := 0 ;
  1010.           UpdateFlag := true ; ;
  1011.         end ;
  1012.  
  1013. { *************************************************************************
  1014.         Select Ascending Sort for the current field.
  1015. ************************************************************************* }
  1016.       procedure ShiftUpArrow(ScrRec : ScrPtr ; DataRec : DataPtr) ;
  1017.  
  1018.         var
  1019.            CurRec     : ScrPtr ;
  1020.            DisplayStr : Str255 ;
  1021.            i          : short_integer ;
  1022.            NewChar    : StrChar ;
  1023.            CharInt    : short_integer ;
  1024.  
  1025.          begin
  1026.            GetStr(DataRec, DisplayStr, ScrRec^.Offset, ScrRec^.Size ) ;
  1027.            IF (ScrRec^.DataType='F') AND (LENGTH(DisplayStr)=1) THEN
  1028.                 Press_BS ;
  1029. { * }      if (Length(DisplayStr) < 1) 
  1030.                 OR 
  1031.               ((LENGTH(DisplayStr)=1) AND (ScrRec^.DataType='F')) then
  1032.               begin
  1033.                 LoByte := 3 ;
  1034.                 CharInInput(LoByte) ;
  1035.                 
  1036.                 CurRec := S_FirstRec[ScrNum] ;
  1037.                 i := 1 ;
  1038.                 While CurRec <> nil do
  1039.                    begin
  1040.                      if CurRec = ScrRec then
  1041.                         CurRec := nil
  1042.                      else
  1043.                         begin
  1044.                           CurRec := CurRec^.Next ;
  1045.                           i := i + 1 ;
  1046.                         end ;
  1047.                    end ;
  1048.                    
  1049.                 Int_AddARec(F_FirstRec,F_CurRec,F_LastRec, i) ;
  1050.                 WriteV(DisplayStr, SortCount) ;
  1051.                 for i := 1 to Length(DisplayStr) do
  1052.                     begin
  1053.                       NewChar := Copy(DisplayStr,1,1) ;
  1054.                       GetAscii(NewChar, CharInt) ;
  1055.                       LoByte := CharInt ;
  1056.                       CharInInput(LoByte) ;
  1057.                       Delete(DisplayStr,1,1) ;
  1058.                     end ;
  1059.                 SortCount := SortCount + 1 ;
  1060.               end ;
  1061.          end ;
  1062.  
  1063. { *************************************************************************
  1064.         Select descending sort for the current field.
  1065. ************************************************************************* }
  1066.       procedure ShiftDownArrow(ScrRec : ScrPtr ; DataRec : DataPtr) ;
  1067.  
  1068.         var
  1069.            CurRec     : ScrPtr ;
  1070.            DisplayStr : Str255 ;
  1071.            i          : short_integer ;
  1072.            NewChar    : StrChar ;
  1073.            CharInt    : short_integer ;
  1074.  
  1075.          begin
  1076.            GetStr(DataRec, DisplayStr, ScrRec^.Offset, ScrRec^.Size ) ;
  1077.            IF (ScrRec^.DataType='F') AND (LENGTH(DisplayStr)=1) THEN
  1078.                 Press_BS ;
  1079. { * }      if (Length(DisplayStr) < 1) 
  1080.                 OR 
  1081.               ((LENGTH(DisplayStr)=1) AND (ScrRec^.DataType='F')) then
  1082.               begin
  1083.                 LoByte := 2 ;
  1084.                 CharInInput(LoByte) ;
  1085.  
  1086.                 CurRec := S_FirstRec[ScrNum] ;
  1087.                 i := 1 ;
  1088.                 While CurRec <> nil do
  1089.                    begin
  1090.                      if CurRec = ScrRec then
  1091.                         CurRec := nil
  1092.                      else
  1093.                         begin
  1094.                           CurRec := CurRec^.Next ;
  1095.                           i := i + 1 ;
  1096.                         end ;
  1097.                    end ;
  1098.  
  1099.                 Int_AddARec(F_FirstRec,F_CurRec,F_LastRec, i) ;
  1100.                 WriteV(DisplayStr, SortCount) ;
  1101.                 for i := 1 to Length(DisplayStr) do
  1102.                     begin
  1103.                       NewChar := Copy(DisplayStr,1,1) ;
  1104.                       GetAscii(NewChar, CharInt) ;
  1105.                       LoByte := CharInt ;
  1106.                       CharInInput(LoByte) ;
  1107.                       Delete(DisplayStr,1,1) ;
  1108.                     end ;
  1109.                 SortCount := SortCount + 1 ;
  1110.               end ;
  1111.          end ;
  1112.  
  1113.        procedure PressUndo ;
  1114.  
  1115.          var
  1116.             CurRec : DataPtr ;
  1117.             
  1118.           begin
  1119.             if DelItem <> nil then
  1120.                begin
  1121.                  TotalRec[DataNum] := TotalRec[DataNum] + 1 ;
  1122.                  if DelItem^.Prev <> nil then
  1123.                     DelItem^.Prev^.Next := DelItem ;
  1124.                  if DelItem^.Next <> nil then
  1125.                     DelItem^.Next^.Prev := DelItem ;
  1126.                  if DelItem^.Prev = D_LastRec[DataNum] then
  1127.                     D_LastRec[DataNum] := DelItem ;
  1128.                  if DelItem^.Next = D_FirstRec[DataNum] then
  1129.                     D_FirstRec[DataNum] := DelItem ;
  1130.                  D_CurrentRec[ScrNum] := DelItem ;
  1131.                  DelItem := nil ;
  1132.                  DrawRecord(D_CurrentRec[DataNum]) ;
  1133.                  
  1134.                  RecNo[DataNum] := 1 ;
  1135.                  CurRec := D_FirstRec[DataNum] ;
  1136.                  While CurRec <> nil do
  1137.                     begin
  1138.                       if CurRec = D_CurrentRec[DataNum] then
  1139.                          CurRec := nil
  1140.                       else
  1141.                          begin
  1142.                            CurRec := CurRec^.Next ;
  1143.                            RecNo[DataNum] := RecNo[DataNum] + 1 ;
  1144.                          end ;
  1145.                     end ;
  1146.  
  1147.                  UpdateFlag := true ; ;
  1148.                end ;
  1149.           end ;
  1150.  
  1151. { *************************************************************************
  1152.         Keyboard parser for input mode : MODE = 2.
  1153. ************************************************************************* }
  1154.       procedure InMode ;
  1155.  
  1156.          begin
  1157.            if (HiByte = $4D) AND (LoByte = $36) then      { Shift-Right }
  1158.               IncrementRec(D_CurrentRec[DataNum], 1, true)
  1159.            else
  1160.               if (HiByte = $4B) AND (LoByte = $34) then   { Shift-Left  }
  1161.                  IncrementRec(D_CurrentRec[DataNum], -1, true)
  1162.               else
  1163.                      case HiByte of
  1164.                          $01 : if LoByte = $1B then           { ESC }
  1165.                                   Press_ESC ;
  1166.                          $0E : if (LoByte = $08) AND          { backspace }
  1167.                                   (S_CurrentRec[ScrNum]^.XInPos - 1 >= 0) then
  1168.                                   Press_BS ;
  1169.                          $0F : if LoByte = $09 then
  1170.                                   Press_Tab ;                 { TAB }
  1171.                          $1C : if LoByte = $0D then           { return }
  1172.                                   Press_Ret ;
  1173.                          $20 : if LoByte = $04 then           { ^D }
  1174.                                   DeleteRecord ;
  1175.  
  1176.                          $53 : if LoByte = $7F then           { delete }
  1177.                                   Press_Del ;
  1178.                          $72 : if LoByte = $0D then           { enter  }
  1179.                                   Press_Ret ;
  1180.                      end ;
  1181.          end ;
  1182.  
  1183. { *************************************************************************
  1184.         Keyboard parser for search mode : MODE = 3.
  1185. ************************************************************************* }
  1186.       procedure SearchMode ;
  1187.  
  1188.          begin
  1189.            if (HiByte = $4B) AND (LoByte = $34) then        { Shift-Left }
  1190.               IncrementRec(D_CurrentRec[DataNum], -1, true)
  1191.            else
  1192.               if (HiByte = $4D) AND (LoByte = $36) then    { Shift-Right }
  1193.                  IncrementRec(D_CurrentRec[DataNum], 1, true)
  1194.               else
  1195.                     case HiByte of
  1196.                        $01 : if LoByte = $1B then           { ESC }
  1197.                                 Press_ESC ;
  1198.                        $0E : if (LoByte = $08) AND          { backspace }
  1199.                                 (S_CurrentRec[ScrNum]^.XInPos - 1 >= 0) then
  1200.                                 Press_BS ;
  1201.                        $1C : if LoByte = $0D then           { return }
  1202.                                 Press_Ret ;
  1203.                        $20 : if LoByte = $04 then           { ^D }
  1204.                                 DeleteRecord ;
  1205.                        $53 : if LoByte = $7F then           { delete }
  1206.                                 Press_Del ;
  1207.                        $72 : if LoByte = $0D then           { enter  }
  1208.                                 Press_Ret ;
  1209.                    end ;
  1210.          end ;
  1211.  
  1212. { *************************************************************************
  1213.         Keyboard parser for input mode : MODE = 4.
  1214. ************************************************************************* }
  1215.       procedure SortMode ;
  1216.  
  1217.          begin
  1218.            if (HiByte = $48) AND (LoByte = $38) then       { Shift-Up    }
  1219.               ShiftUpArrow(S_CurrentRec[ScrNum],D_CurrentRec[DataNum])
  1220.            else
  1221.               if (HiByte = $50) AND (LoByte = $32) then    { Shift-Down  }
  1222.                  ShiftDownArrow(S_CurrentRec[ScrNum],D_CurrentRec[DataNum])
  1223.               else
  1224.                  if (LoByte = $0D) AND 
  1225.                    ((HiByte = $1C) OR (HiByte = $72)) then
  1226.                      Press_Ret ;
  1227.          end ;
  1228.  
  1229.   begin
  1230.       if LoByte = $00 then 
  1231.          begin
  1232.            Case HiByte of
  1233.               $44 : SelectDate(S_CurrentRec[ScrNum], 
  1234.                                D_CurrentRec[ScrNum], DisplayStr) ;
  1235.               $47 : ClrHome ;
  1236.               $48 : UpArrow(S_CurrentRec[ScrNum]) ;
  1237.               $4B : LeftArrow(S_CurrentRec[ScrNum]) ;
  1238.               $4D : RightArrow(S_CurrentRec[ScrNum]) ;
  1239.               $50 : DownArrow(S_CurrentRec[ScrNum]) ;
  1240.               $61 : IF (Mode = 2) OR (Mode=3) THEN PressUndo ;
  1241.               $73 : GoToFirst(D_CurrentRec[DataNum],true) ;
  1242.               $74 : GoToLast(D_CurrentRec[DataNum], true) ;
  1243.              else : if Mode = 3 then
  1244.                        begin
  1245.                           Case HiByte of
  1246.                               $3B : C_CurRec^.Match := 1 ;
  1247.                               $3C : C_CurRec^.Match := 2 ;
  1248.                               $3D : C_CurRec^.Match := 3 ;
  1249.                               $3E : C_CurRec^.Match := 4 ;
  1250.                               $3F : C_CurRec^.Match := 5 ;
  1251.                               $40 : C_CurRec^.Match := 6 ;
  1252.                           end ;
  1253.                          if (HiByte > $3A) AND (HiByte < $41) then
  1254.                             UpdateFlag := true ; ;
  1255.                        end ;
  1256.            end ;
  1257.          end
  1258.       else
  1259.          Case Mode of
  1260.              2 : InMode ;
  1261.              3 : SearchMode ;
  1262.              4 : SortMode ;
  1263.          end ;
  1264.     end;
  1265.  
  1266. { *************************************************************************
  1267.      Keyboard_Input is called by Event_Loop whenever a keyboard event
  1268.      is detected.  Depending on the current mode, one of the scanning
  1269.      modules is called to interpret the keyboard input.
  1270. ************************************************************************* }
  1271.   procedure Keyboard_Input( KeyValue : short_integer);
  1272.  
  1273.     var
  1274.        NewMode,
  1275.        HiByte,
  1276.        LoByte : short_integer ;
  1277.        KeyParse,
  1278.        KeyIn  : boolean ;
  1279.  
  1280.      begin
  1281.        KeyIn   := false ;
  1282.        KeyParse := false ;
  1283.        NewMode := Mode ;
  1284.        HiByte  := ShR(KeyValue, 8);
  1285.        LoByte  := ShR(ShL(KeyValue, 8),8);
  1286.  
  1287.        if Mode = 5 then
  1288.           EraseCursor(Report)
  1289.        else
  1290.           EraseCursor(ScrNum) ;
  1291.  
  1292.        if LoByte = $00 then 
  1293.           Case HiByte of
  1294.               $17 : if ((Mode = 1) AND (S_FirstRec[ScrNum] <> nil)) OR
  1295.                         (Mode = 2) OR (Mode = 3) OR (Mode = 4) OR
  1296.                        ((Mode = 5) AND (D_FirstRec[DataNum] <> nil) AND 
  1297.                                        (F_FirstRec = nil)) then
  1298.                        SelectInput(NewMode) ;    { alt-I : Input }
  1299.               $18 : if ((Mode = 2) OR (Mode = 3) OR (Mode=5)) AND
  1300.                        (TotalRec[DataNum] > 1) then
  1301.                        SelectOutput(NewMode) ;   { alt-O : Output }
  1302.               $20 : if  (Mode = 2) OR 
  1303.                        ((Mode = 5) AND (D_FirstRec[DataNum] = nil) AND 
  1304.                                        (F_FirstRec = nil)) then
  1305.                        NewMode := 1 ;            { alt-D : Design }
  1306.               $21 : if ((Mode = 2) OR (Mode = 3) OR
  1307.                        ((Mode = 5) AND (F_FirstRec <> nil))) AND
  1308.                        (TotalRec[DataNum] > 1) then
  1309.                        SelectSearch(NewMode) ;   { alt-F : Search }
  1310.               $1F : if ((Mode = 2) OR (Mode = 4)) AND
  1311.                         (TotalRec[DataNum] > 1) then
  1312.                        SelectSort(NewMode) ;     { alt-S : Sort   }
  1313.               $62 : HelpScreen ;
  1314.              else : KeyParse := true ;
  1315.           end
  1316.        else
  1317.           if ((LoByte > $1F) AND (LoByte < $7F)) AND
  1318.              ((HiByte < $40) OR  (HiByte > $62) OR 
  1319.               (HiByte = $4A) OR  (HiByte = $4E)) then
  1320.              Case Mode of
  1321.               2,3 : if S_CurrentRec[ScrNum]^.XInPos < 
  1322.                        S_CurrentRec[ScrNum]^.Size then
  1323.                        begin
  1324.                          KeyIn := true ;
  1325.                          CharInInput(LoByte) ;
  1326.                        end ;
  1327.                 5 : if ((XCur < 80) AND NOT PrtFlag[1]) OR
  1328.                        ((XCur < 132) AND PrtFlag[1]) then
  1329.                        begin 
  1330.                          CharRPInput(LoByte) ;
  1331.                          if (XCur > 75) AND (RW_Offset = 0) then
  1332.                             begin
  1333.                               RW_Offset := 57 ;
  1334.                               DrawDZ_Out ;
  1335.                            end ;
  1336.                        end ;
  1337.              end
  1338.           else
  1339.              KeyParse := true ;
  1340.        
  1341.        if KeyParse then
  1342.           Case HiByte of
  1343. { ^Q }       $10 : if (LoByte = $11) AND (Mode <> 3) AND (Mode <> 4) then
  1344.                       ExitProgram
  1345.                    else
  1346.                       KeyParse := true ;
  1347. { ^O }       $18 : if (LoByte = $0F) AND ((Mode = 1) OR (Mode = 2)) then
  1348.                       Select_Open(NewMode)
  1349.                    else
  1350.                       KeyParse := true ;
  1351. { ^S }       $1F : if (LoByte = $13) AND
  1352.                       (((Mode = 1) AND (S_FirstRec[ScrNum] <> nil)) OR
  1353.                        ((Mode = 3) AND NOT (SearchFlag)) OR
  1354.                        ((Mode = 2)) AND (TotalRec[DataNum] > 1)) then
  1355.                         Select_Save
  1356.                    else
  1357.                       KeyParse := true ;
  1358. { ^C }       $2E : if (LoByte = $03) AND
  1359.                       (((Mode = 1) AND (S_FirstRec[ScrNum] <> nil)) OR
  1360.                         (Mode = 2)) then
  1361.                       Select_Close
  1362.                    else
  1363.                       KeyParse := true ;
  1364.           end ;
  1365.  
  1366.        if KeyParse then
  1367.           Case Mode of
  1368.               1 : KB_InDesign( HiByte, LoByte, NewMode, KeyIn );
  1369.              2,3,
  1370.               4 : KB_InInput( HiByte, LoByte, NewMode, KeyIn );
  1371.               5 : KB_InReport( HiByte, LoByte, NewMode, KeyIn );
  1372.           end ;    
  1373.  
  1374.        if NOT KeyIn then
  1375.           begin
  1376.             if Mode <> NewMode then
  1377.                ChangeMode(Mode, NewMode) ;
  1378.             MenuOption ;
  1379.           end ;
  1380.  
  1381.        if WindNum > 0 then
  1382.           begin
  1383.             if Mode = 5 then
  1384.                NewCursor(Report)
  1385.             else
  1386.                NewCursor(ScrNum) ;
  1387.           end ;
  1388.      end ;
  1389.  
  1390. { *************************************************************************
  1391.      MB_InDesign is called by MB_Input below whenever a mouse button event
  1392.      is detected while in the Design mode.  This procedure evaluates a 
  1393.      left button event to relocate the position of the cursor.  The old 
  1394.      cursor position is redrawn at the beginning of the procedure and 
  1395.      the new cursor position drawn at the end of the procedure.
  1396. ************************************************************************* }
  1397.   procedure MB_InDesign( M_XPos, M_YPos : short_integer );
  1398.  
  1399.    var
  1400.       XTemp,
  1401.       YTemp : short_integer ;
  1402.  
  1403.     begin
  1404.       XTemp := (M_XPos - x) DIV 8 ;
  1405.       if (XTemp > 0) AND (XTemp < w DIV 8 - 1) then
  1406.          XCur := XTemp ;   { 7 }
  1407.       YTemp := (M_YPos - y + 9 * Resolution) DIV Spacing ;
  1408.       if (YTemp > 0) AND (YTemp < h DIV Spacing + 1) then
  1409.          YCur := YTemp ;
  1410.     end;
  1411.  
  1412. { *************************************************************************
  1413.      MB_InReport is called by MB_Input below whenever a mouse button event
  1414.      is detected while in the Output mode.  This procedure evaluates a 
  1415.      left button event to relocate the position of the cursor.  The old 
  1416.      cursor position is redrawn at the beginning of the procedure and 
  1417.      the new cursor position drawn at the end of the procedure.
  1418. ************************************************************************* }
  1419.   procedure MB_InReport( M_XPos, M_YPos : short_integer );
  1420.  
  1421.    var
  1422.       Location,
  1423.       i,
  1424.       CurLoc,
  1425.       Counter,
  1426.       XTemp,
  1427.       YTemp   : short_integer ;
  1428.       ScrRec  : ScrPtr ;
  1429.       NewChar : StrChar ;
  1430.  
  1431.       Start,
  1432.       Count   : short_integer ;
  1433.  
  1434.     begin
  1435.       if M_YPos > y + h DIV 2 - 23 * Resolution then
  1436.          begin
  1437.            XTemp := (M_XPos - x) DIV 8 ;
  1438.            if XTemp < 1 then XTemp := 1
  1439.            else
  1440.               if XTemp > 76 then XTemp := 76 ;
  1441.            XCur := XTemp + RW_Offset ;
  1442.  
  1443.            YTemp := (M_YPos - y + 12 * Resolution) DIV Spacing ;
  1444.            if YTemp < 8 then YTemp := 8
  1445.            else
  1446.               if YTemp > 17 then YTemp := 17 ;
  1447.            YCur := YTemp ;
  1448.            if XCur > RepWidth then XCur := RepWidth ;
  1449.  
  1450.            CheckCurLoc(CurLoc, ScrRec, XCur, YCur, Report ) ;
  1451.            S_CurrentRec[Report] := ScrRec ;
  1452.          end
  1453.       else
  1454.         if M_YPos > 29 * Resolution then
  1455.         begin
  1456.           Case RepLine of
  1457.              1 : begin
  1458.                    Start := 5 ;
  1459.                    Count := 2 ;
  1460.                  end ;
  1461.              2 : begin
  1462.                    Start := 5 ;
  1463.                    Count := 3 ;
  1464.                  end ;
  1465.              3 : begin
  1466.                    Start := 5 ;
  1467.                    Count := 4 ;
  1468.                  end ;
  1469.              4 : begin
  1470.                    Start := 4 ;
  1471.                    Count := 5 ;
  1472.                  end ;
  1473.           end ;
  1474.  
  1475.         if ((P_Mode = 2) AND
  1476.             (YCur > 7 + Start) AND (YCur < 7 + Start + Count)) OR
  1477.             (P_Mode = 1) OR (P_Mode = 0) then
  1478.          begin
  1479.            R_EditFlag := true ;
  1480.            if (M_XPos > 71) AND (M_XPos < 275) then XTemp := 0
  1481.            else
  1482.               if (M_XPos > 350) AND (M_XPos < 556) then XTemp := 1
  1483.                  else XTemp := 2 ;
  1484.            if XTemp < 2 then
  1485.               begin
  1486.                 YTemp := (M_YPos - y - 2 * Resolution) DIV 
  1487.                          (12 * Resolution) ;
  1488.                 if YTemp < 0 then YTemp := 0 ;
  1489.                 if YTemp > 4 then YTemp := 4 ;
  1490.                 Counter := YTemp + PL_Offset + (XTemp * 5) ;
  1491.                 ScrRec := S_FirstRec[ScrNum] ;
  1492.                 for i := 1 to Counter do
  1493.                     begin
  1494.                       ScrRec := ScrRec^.Next ;
  1495.                       if ScrRec = nil then i := Counter + 1 ;
  1496.                     end ;
  1497.     
  1498.                 if ScrRec <> nil then
  1499.                    begin
  1500.                      if XCur + ScrRec^.Size < RepWidth + 2 then
  1501.                         begin
  1502.                           FormatStr := '' ;
  1503.                           for i := XCur - 1 to XCur + ScrRec^.Size - 2 do
  1504.                               begin
  1505.                                 Location := S_CurrentRec[Report]^.Offset + i ;
  1506.                                 ModifyStr(D_CurrentRec[Report], Location, 
  1507.                                           chr(Counter + $80)) ;
  1508.                                 FormatStr := Concat(FormatStr, chr(Counter + $41) ) ;
  1509.                               end ;
  1510.                           Draw_String(x + (XCur - RW_Offset) * 8,
  1511.                                       y + YCur * Spacing - 4 * Resolution, 
  1512.                                       FormatStr) ;
  1513.                         end
  1514.                      else
  1515.                         begin
  1516.                           AlertStr := '[2][Insufficient Room for | |' ;
  1517.                           AlertStr := Concat(AlertStr, '   Selected Field | ]') ;
  1518.                           AlertStr := Concat(AlertStr, '[ Continue ]') ;
  1519.                           Result   := Do_Alert(AlertStr,1) ;
  1520.                         end ;
  1521.                    end ;
  1522.               end ;
  1523.          end ;
  1524.         end ;
  1525.     end;
  1526.  
  1527. { *************************************************************************
  1528.      MB_InInput is called by MB_Input whenever a mouse event is detected
  1529.      while the program is in the Input mode.  The mouse position is
  1530.      checked to see if it is in a legitimate input box and relocates the
  1531.      cursor if the position is valid.  The old cursor is erased and the
  1532.      new cursor draw.
  1533. ************************************************************************* }
  1534.   procedure MB_InInput( M_XPos, M_YPos : short_integer );
  1535.  
  1536.    Var
  1537.       XTemp,
  1538.       YTemp   : short_integer ;
  1539.       CurLoc  : short_integer ;
  1540.       CurRec  : ScrPtr ;
  1541.       NewChar : StrChar ;
  1542.  
  1543.     begin
  1544.       XTemp := (M_XPos - x) DIV 8 ;
  1545.       YTemp := (M_YPos - y + 8 * Resolution) DIV Spacing ;
  1546. {
  1547.       if (M_YPos > 18 * Resolution) AND (M_YPos < 28 * Resolution) AND
  1548.          (M_XPos > 4) AND (M_XPos < 388) AND SearchFlag then
  1549.          C_CurRec^.Match := (M_XPos + 60) DIV 64 ;
  1550. }
  1551.       CheckCurLoc(CurLoc, CurRec, XTemp, YTemp, ScrNum ) ;
  1552.       if CurLoc > -1 then
  1553.          GetChar(CurRec, D_CurrentRec[DataNum], NewChar, CurRec^.Offset) 
  1554.       else
  1555.          NewChar := chr(2) ;
  1556.  
  1557.       if NewChar = chr(1) then
  1558.          begin
  1559.            XCur := CurRec^.XPos ;
  1560.            YCur := YTemp ;
  1561.            S_CurrentRec[ScrNum]^.XInPos := 0 ;
  1562.            S_CurrentRec[ScrNum] := CurRec ;
  1563.            S_CurrentRec[ScrNum]^.XInPos := 0 ;
  1564.          end
  1565.       else
  1566.          Repeat
  1567.            CheckCurLoc(CurLoc, CurRec, XTemp, YTemp, ScrNum ) ;
  1568.            if CurLoc > -1 then
  1569.               begin
  1570.                 if (CurRec^.X + CurLoc >= CurRec^.XPos) AND 
  1571.                    (CurRec^.X + CurLoc <  CurRec^.XPos + CurRec^.Size) then
  1572.                    begin
  1573.                      GetChar(CurRec, D_CurrentRec[DataNum], NewChar, 
  1574.                              CurRec^.Offset + CurLoc - CurRec^.XPos + CurRec^.X) ;
  1575.                      if NewChar <> chr(1) then
  1576.                         begin
  1577.                           S_CurrentRec[ScrNum] := CurRec ;
  1578.                           CurRec^.XInPos := CurLoc + CurRec^.X - CurRec^.XPos ; 
  1579.                           XCur := XTemp ;
  1580.                           YCur := YTemp ;
  1581.                           CurLoc := -1 ;
  1582.                         end
  1583.                      else
  1584.                         XTemp := XTemp - 1 ;
  1585.                    end
  1586.                 else
  1587.                    if CurLoc < CurRec^.XPos - CurRec^.X then
  1588.                       XTemp := XTemp + 1
  1589.                    else
  1590.                       XTemp := XTemp - 1 ;
  1591.               end 
  1592.            else
  1593.               CurLoc := -1;
  1594.  
  1595.          Until CurLoc < 0 ;
  1596.     end;
  1597.  
  1598. { *************************************************************************
  1599.      MB_Input is called by Event_Loop in the Main program module.  This
  1600.      procedure decides which of the decision procedures to call depending
  1601.      upon the current program mode.
  1602. ************************************************************************* }
  1603.   procedure MB_Input( M_XPos, M_YPos : short_integer );
  1604.  
  1605.      begin
  1606.        Hide_Mouse ;
  1607.        if Mode = 5 then
  1608.           EraseCursor(Report)
  1609.        else
  1610.           EraseCursor(ScrNum) ;
  1611.  
  1612.        Case Mode of
  1613.             1 : MB_InDesign( M_XPos, M_YPos );
  1614.           2,3,
  1615.           4,6 : MB_InInput( M_XPos, M_YPos );
  1616.             5 : MB_InReport( M_XPos, M_YPos );
  1617.        end ;    
  1618.  
  1619.        if WindNum > 0 then
  1620.           begin
  1621.             if Mode = 5 then
  1622.                NewCursor(Report)
  1623.             else
  1624.                NewCursor(ScrNum) ;
  1625.           end ;
  1626.  
  1627.       MenuOption ;
  1628.       UpdateFlag := true ; ;
  1629.       Show_Mouse ;
  1630.      end ;
  1631.  
  1632. BEGIN
  1633. END .
  1634.  
  1635.  
  1636.